# **Project 2: RISC-V Simulator**

Instructor: Prof. Seokin Hong (seokin@knu.ac.kr)

Assigned: November 20, 2019

Due: 11:59pm December 18, 2019

#### 1. Description

The goal of this project is to write a simulator program that simulates the RISC-V microarchitecture (single cycle execution). You need to model the basic operations of the RISC-V microarchitecture, such as fetching instructions from the instruction memory, decoding instructions, executing the proper arithmetic operations, reading data from the register file or the data memory, and determining the address of the next instructions. Your simulator will read an executable file (e.g., runme.hex) that contains several instructions (represented in hexadecimal).

The simulator should have two modes: debug and run mode. In the debug mode, the simulator will show the clock cycles, PC, and register values after executing an instruction every cycle. In the run mode, your simulator will run the given executable file and show the clock cycles, PC, and register values at the end of the simulation.

You can download a zip file (pj2.zip) that includes skeleton code (riscv\_sim.c) and several executable files (addi\_test.hex, add\_test.hex, jal\_test.hex, jalr\_test.hex, beq\_test.hex, sd\_ld\_test.hex, and runme.hex) from the project section of the LMS. You can use the skeleton code to write your simulator program. The files named "(\*\_test.hex)" will be useful for testing your simulator to ensure that it can execute the individual instruction correctly.

Please upload the "riscv\_sim.c" file and a "report" to the LMS system (assignment section). In the report, you need to explain your code briefly.

#### 2. Instruction you must simulate

add, addi, jal, jalr, ld, sd, beq

#### 3. How to run

```
./riscv_sim ./runme.hex 0 // debug mode
./riscv_sim ./runme.hex 1 // run mode
```

## 4. Expected output of your simulator

## a. Debug mode

\* All numbers are decimal number

```
$./riscv_sim runme.hex 0
Clock cycles = 1
PC
       = ?
x0
    = 0
x1
    = ?
x2
x3
x4
x5
x6
    = 0
x7
    = 0
x8
x9
x10
    = ?
    = 0
x11
    = 0
x12
    = 0
x13
    = 0
x14
x15 = 0
x16 = 0
x17 = 0
x18 = 0
x19 = 0
x20 = 0
x21 = 0
x22 = 0
x23 = 0
```

```
x24 = 0
x25 = 0
x26 = 0
x27 = 0
x28 = 0
x29 = 0
x30 = 0
x31 = 0
Clock cycles = 2
PC = ?
x0 = 0
x1 = ?
x2 = ?
x3 = 0
x4 = 0
x5 = ?
x6 = ?
x7 = 0
x8 = 0
x9 = ?
x10 = ?
x11 = 0
x12 = 0
x13 = 0
x14 = 0
x15 = 0
x16 = 0
x17 = 0
x18 = 0
x19 = 0
x20 = 0
x21 = 0
\mathbf{x22} = 0
\mathbf{x23} = 0
x24 = 0
x25 = 0
x26 = 0
x27 = 0
x28 = 0
x29 = 0
x30 = 0
x31 = 0
Clock cycles = 3
PC = ?
x0 = 0
x1 = ?
x2 = ?
x3
    = 0
x4
    = 0
x5
    = ?
x6
    = ?
x7 = 0
```

```
= 0
x8
x9
      = ?
x10 = ?
x11 = 0

\begin{array}{rcl}
\mathbf{x12} & = & 0 \\
\mathbf{x13} & = & 0
\end{array}

x14 = 0
x15 = 0
x16 = 0
x17 = 0
x18 = 0
x19 = 0
x20 = 0
x21 = 0
x22 = 0
x23 = 0
x24 = 0
x25 = 0
x26 = 0
x27 = 0
x28 = 0
x29 = 0
x30 = 0
x31 = 0
• • • • • • • • •
```

#### b. run mode

\* clock cycle is decimal number

```
$./riscv_sim runme.hex 1
Clock cycles = ?
PC
      = ?
x0
   = 0
x1 = 12
x2 = 800
x3 = 0
x4
   = 0
x5
   = 0
x6 = 45
    = 0
x7
x8 = 0
x9 = 10
x10 = 55
x11 = 0
x12 = 0
x13 = 0
x14
x15
   = 0
x16 = 0
x17 = 0
x18 = 0
x19 = 0
x20 = 0
x21 = 0
```

```
x22 = 0

x23 = 0

x24 = 0

x25 = 0

x26 = 0

x27 = 0

x28 = 0

x29 = 0

x30 = 0

x31 = 0
```

#### 5. Testing

You can find several executable files ("\*\_test.hex") that you can use to test your simulator.

- Expected output of your simulator for each executable file
  - a. addi\_test.hex

```
$./riscv_sim addi_test.hex 1
Clock\ cycles = 32
       = 128
    = 0
x0
    = 10
x1
    = 10
x2
    = 10
x3
    = 10
x4
    = 10
x5
   = 10
x6
   = 10
x7
   = 10
x8
   = 10
x9
x10 = 10
x11 = 10
x12
    = 10
x13
    = 10
x14
    = 10
x15
    = 10
x16
    = 10
    = 10
x17
x18
    = 10
    = 10
x19
x20
    = 10
x21
    = 10
x22
    = 10
x23
    = 10
x24
    = 10
x25
    = 10
x26
    = 10
x27
    = 10
x28
    = 10
x29
    = 10
x30
    = 10
    = 10
x31
```

#### b. add\_test.hex

```
$./riscv_sim add_test.hex 1
Clock\ cycles = 4
PC
        = 16
x0
    = 0
x1
    = 0
x2
x3
    = 0
    = 0
x5
    = 0
    = 0
x5 = 0

x6 = 0

x7 = 0
x8 = 0
x9 = 10
x10 = 0
x11 = 0
x12 = 0
x13 = 0
x14
    = 0
x15 = 0
x16 = 10
x17 = 20
x18 = 30
x19 = 0
x20 = 0
x21 = 0
x22 = 0
x23 = 0
x24 = 0
x25 = 0
x26 = 0
x27
    = 0
x28
    = 0
x29
    = 0
    = 0
x30
x31 = 0
```

## c. jal\_test.hex

```
$./riscv_sim jal_test.hex 1
Clock\ cycles = 3
PC
    = 16
x0
     = 0
x1
    = 0
x2
    = 0
x3
x4
    = 0
    = 0
x5
   = 0
x6
x7
   = 0
x8
x9 = 10
x10 = 0
x11 = 0
    = 0
x12
    = 0
x13
x14
    = 0
x15
    = 0
x16
    = 0
x17 = 20
x18
    = 0
x19
    = 0
x20
    = 0
x21
    = 0
x22
    = 0
x23
    = 0
x24
    = 0
x25
    = 0
x26
    = 0
x27
    = 0
x28
    = 0
x29
    = 0
x30
    = 0
x31 = 0
```

## d. jalr\_test.hex

```
$./riscv_sim jalr_test.hex 1
Clock\ cycles = 7
PC
    = 36
x0
     = 0
x1
    = 0
x2
    = 0
     = 0
x3
x4
    = 0
    = 0
x5
   = 0
x6
x7
   = 0
x8
x9 = 10
x10 = 0
x11 = 0
    = 0
x12
    = 0
x13
x14
    = 0
x15
    = 0
x16 = 10
x17 = 20
x18 = 0
x19
    = 20
x20
    = 0
x21
    = 0
x22
    = 0
x23
    = 0
x24
    = 0
x25
    = 0
x26
    = 0
x27
    = 0
x28
    = 0
x29
    = 0
x30
    = 0
x31 = 4
```

#### e. sd\_ld\_test.hex

```
$./riscv_sim sd_ld_test.hex 1
Clock cycles = 5
PC = 20
x0 = 0
x1 = 0
x1 = 0

x2 = 0

x3 = 0

x4 = 30

x5 = 0

x6 = 0

x7 = 0
x = 0
x = 0
x = 0
x = 0
x = 0
x = 0
x11 = 0
x12 = 0
x13 = 0
x14 = 0
x15 = 0
x16 = 30
x17 = 0
x18 = 0
x19 = 0
x20 = 0
x21 = 0
x22 = 0
x23 = 0
x24 = 0
x25 = 0
x26 = 0
x27 = 0
x28 = 0
x29 = 0
x30 = 0
x31 = 0
```

#### f. beq\_test.hex

```
$./riscv_sim beq_test.hex 1
Clock\ cycles = 4
PC
    = 20
x0
     = 0
x1
    = 0
x2
    = 0
     = 0
x3
x4
    = 0
    = 0
x5
   = 0
= 0
x6
x7
   = 0
x8
x9 = 10
x10 = 0
x11 = 0
    = 0
x12
    = 0
x13
x14
    = 0
x15
    = 0
x16
    = 30
x17 = 30
x18 = 0
x19
    = 0
x20
    = 0
x21
    = 0
x22
    = 0
x23
    = 0
x24
    = 0
x25
    = 0
x26
    = 0
x27
    = 0
x28
    = 0
x29
    = 0
x30
     = 0
x31 = 0
```

#### **6. Evaluation** [total: 100 points]

- **a.** Unit test [45 points]: test your simulator to ensure that it can execute individual instruction (add, addi, jal, jalr, ld, sd, beq) correctly.
- **b. Full test** [55 points]: test your simulator to ensure that it can correctly execute the given executable file. We will use the "runme.hex" file for the full test.

#### **Late Day Policy**

This project is due at 11:59pm on the due date. A grading penalty will be applied to late assignments. Any assignment turned in late will be penalized 25% per late day.

### Plagiarism

<u>No plagiarism will be tolerated</u>. If the instructor determines that there are substantial similarities between your code and others, <u>you will be given 0 points for all project assignments</u>. Code similarity will be checked with a plagiarism detector tool.

## Appendix A: RISC-V Instructions

| Category      | Instruction                          | Example           | Meaning                       | Comments                                                   |
|---------------|--------------------------------------|-------------------|-------------------------------|------------------------------------------------------------|
|               | Add                                  | add x5, x6, x7    | x5 = x6 + x7                  | Three register operands; add                               |
| Arithmetic    | Subtract                             | sub x5, x6, x7    | x5 = x6 - x7                  | Three register operands; subtract                          |
|               | Add immediate                        | addi x5, x6, 20   | x5 = x6 + 20                  | Used to add constants                                      |
|               | Load doubleword                      | ld x5, 40(x6)     | x5 = Memory[x6 + 40]          | Doubleword from memory to register                         |
|               | Store doubleword                     | sd x5, 40(x6)     | Memory[x6 + 40] = x5          | Doubleword from register to memory                         |
|               | Load word                            | lw x5, 40(x6)     | x5 = Memory[x6 + 40]          | Word from memory to register                               |
|               | Load word, unsigned                  | 1wu x5, 40(x6)    | x5 = Memory[x6 + 40]          | Unsigned word from memory to registe                       |
|               | Store word                           | sw x5, 40(x6)     | Memory[x6 + 40] = x5          | Word from register to memory                               |
|               | Load halfword                        | 1h x5, 40(x6)     | x5 = Memory[x6 + 40]          | Halfword from memory to register                           |
| Data transfer | Load halfword,<br>unsigned           | 1hu x5, 40(x6)    | x5 = Memory[x6 + 40]          | Unsigned halfword from memory to register                  |
|               | Store halfword                       | sh x5, 40(x6)     | Memory[x6 + 40] = x5          | Halfword from register to memory                           |
|               | Load byte                            | 1b x5, 40(x6)     | x5 = Memory[x6 + 40]          | Byte from memory to register                               |
|               | Load byte, unsigned                  | 1bu x5, 40(x6)    | x5 = Memory[x6 + 40]          | Byte unsigned from memory to registe                       |
|               | Store byte                           | sb x5, 40(x6)     | Memory[x6 + 40] = x5          | Byte from register to memory                               |
|               | Load reserved                        | 1r.d x5, (x6)     | x5 = Memory[x6]               | Load; 1st half of atomic swap                              |
|               | Store conditional                    | sc.d x7, x5, (x6) | Memory[x6] = x5; $x7 = 0/1$   | Store; 2nd half of atomic swap                             |
|               | Load upper immediate                 | lui x5, 0x12345   | x5 = 0x12345000               | Loads 20-bit constant shifted left 12 bits                 |
|               | And                                  | and x5, x6, x7    | x5 = x6 & x7                  | Three reg. operands; bit-by-bit AND                        |
|               | Inclusive or                         | or x5, x6, x8     | x5 = x6   x8                  | Three reg. operands; bit-by-bit OR                         |
| orical        | Exclusive or                         | xor x5, x6, x9    | x5 = x6 ^ x9                  | Three reg. operands; bit-by-bit XOR                        |
| _ogical       | And immediate                        | andi x5, x6, 20   | x5 = x6 & 20                  | Bit-by-bit AND reg. with constant                          |
|               | Inclusive or immediate               | ori x5, x6, 20    | $x5 = x6 \mid 20$             | Bit-by-bit OR reg. with constant                           |
|               | Exclusive or immediate               | xori x5, x6, 20   | $x5 = x6 ^20$                 | Bit-by-bit XOR reg. with constant                          |
|               | Shift left logical                   | sll x5, x6, x7    | x5 = x6  << x7                | Shift left by register                                     |
|               | Shift right logical                  | srl x5, x6, x7    | $x5 = x6 \gg x7$              | Shift right by register                                    |
|               | Shift right arithmetic               | sra x5, x6, x7    | $x5 = x6 \gg x7$              | Arithmetic shift right by register                         |
| Shift         | Shift left logical immediate         | slli x5, x6, 3    | x5 = x6 << 3                  | Shift left by immediate                                    |
|               | Shift right logical immediate        | srli x5, x6, 3    | x5 = x6 >> 3                  | Shift right by immediate                                   |
|               | Shift right arithmetic immediate     | srai x5, x6, 3    | x5 = x6 >> 3                  | Arithmetic shift right by immediate                        |
|               | Branch if equal                      | beg x5, x6, 100   | if (x5 == x6) go to PC+100    | PC-relative branch if registers equal                      |
|               | Branch if not equal                  | bne x5, x6, 100   | if (x5 != x6) go to PC+100    | PC-relative branch if registers not equ                    |
|               | Branch if less than                  | blt x5, x6, 100   | if (x5 < x6) go to PC+100     | PC-relative branch if registers less                       |
| Conditional   | Branch if greater or equal           | bge x5, x6, 100   | if $(x5 \ge x6)$ go to PC+100 | PC-relative branch if registers greater or equal           |
| oranch        | Branch if less, unsigned             | bltu x5, x6, 100  | if (x5 < x6) go to PC+100     | PC-relative branch if registers less, unsigned             |
|               | Branch if greater or equal, unsigned | bgeu x5, x6, 100  | if (x5 >= x6) go to PC+100    | PC-relative branch if registers greater or equal, unsigned |
| Inconditional | Jump and link                        | jal x1, 100       | x1 = PC+4; go to $PC+100$     | PC-relative procedure call                                 |
| ranch         | Jump and link register               | jalr x1, 100(x5)  | x1 = PC+4; go to $x5+100$     | Procedure return; indirect call                            |

## Appendix B: RISC-V Encoding

| Name         |                             | Fi              |        | Comments |               |        |                               |
|--------------|-----------------------------|-----------------|--------|----------|---------------|--------|-------------------------------|
| (Field size) | 7 bits                      | 5 bits          | 5 bits | 3 bits   | 5 bits        | 7 bits |                               |
| R-type       | funct7                      | rs2             | rs1    | funct3   | rd            | opcode | Arithmetic instruction format |
| I-type       | immediate                   | immediate[11:0] |        | funct3   | rd            | opcode | Loads & immediate arithmetic  |
| S-type       | immed[11:5]                 | rs2             | rs1    | funct3   | immed[4:0]    | opcode | Stores                        |
| SB-type      | immed[12,10:5]              | rs2             | rs1    | funct3   | immed[4:1,11] | opcode | Conditional branch format     |
| UJ-type      | immediate[20,10:1,11,19:12] |                 |        |          | rd            | opcode | Unconditional jump format     |
| U-type       | immediate[31:12]            |                 |        |          | rd            | opcode | Upper immediate format        |

#### 1) R-type: add

a. add rd, rs1, rs2

| 31                       | 25 | 24                    | 20 19 | 1                     | 5 14         |         | 12 11         | 7                     | 7 6 |       | 0 |
|--------------------------|----|-----------------------|-------|-----------------------|--------------|---------|---------------|-----------------------|-----|-------|---|
| $\operatorname{funct} 7$ |    | rs2                   |       | rs1                   |              | funct3  |               | $^{\mathrm{rd}}$      | C   | pcode |   |
| 7                        |    | 5                     |       | 5                     |              | 3       |               | 5                     |     | 7     |   |
| 0000000                  |    | ${ m src2}$           |       | $\operatorname{src}1$ | ADD          | SLT/SL  | $\mathrm{TU}$ | $\operatorname{dest}$ |     | OP    |   |
| 0000000                  |    | $\operatorname{src2}$ |       | src1                  | ANI          | O/OR/XO | $\mathbf{R}$  | $\operatorname{dest}$ |     | OP    |   |
| 0000000                  |    | ${ m src2}$           |       | $\operatorname{src}1$ | $\mathbf{S}$ | LL/SRL  |               | $\operatorname{dest}$ |     | OP    |   |
| 0100000                  |    | ${ m src2}$           |       | $\operatorname{src}1$ | $S^{1}$      | UB/SRA  |               | $\operatorname{dest}$ |     | OP    |   |

## 2) I-type: addi, ld, jalr

- a. addi rd, rs1, imm
- **b.** ld rd, imm(rs1)
- **c.** jalr rd, imm(rs1)

| 31                          | 20 19                | 15 14 | 12                      | 2 11                  | 7 6    | 0 |
|-----------------------------|----------------------|-------|-------------------------|-----------------------|--------|---|
| $\mathrm{imm}[11:0]$        | rs1                  |       | $\operatorname{funct3}$ | rd                    | opcode |   |
| 12                          | 5                    |       | 3                       | 5                     | 7      |   |
| I-immediate[11:0]           | $\operatorname{src}$ | AI    | DDI/SLTI[U]             | $\operatorname{dest}$ | OP-IMM |   |
| $\hbox{I-immediate} [11:0]$ | $\operatorname{src}$ | AN    | DI/ORI/XO               | RI dest               | OP-IMM |   |

#### 3) s-type: sd

a. sd rs2, imm(rs1)

| 31 | 1 25         | 24 20                | 19 15 | 14 12  | 11 7        | 7 6    | 0 |
|----|--------------|----------------------|-------|--------|-------------|--------|---|
|    | imm[11:5]    | rs2                  | rs1   | funct3 | imm[4:0]    | opcode |   |
|    | 7            | 5                    | 5     | 3      | 5           | 7      |   |
|    | offset[11:5] | $\operatorname{src}$ | base  | width  | offset[4:0] | STORE  |   |

## 4) SB-type: beq

a. beq rs1, rs2, imm

| 31 30         | 25 24       | 20 19 1               | 5 14 12 | 2 11 8    | 3 7     | 6      | 0 |
|---------------|-------------|-----------------------|---------|-----------|---------|--------|---|
| imm[12] imr   | n[10:5] rs2 | rs1                   | funct3  | imm[4:1]  | imm[11] | opcode |   |
| 1             | 6 5         | 5                     | 3       | 4         | 1       | 7      |   |
| offset[12,10] | ):5] src2   | $\operatorname{src}1$ | BEQ/BNE | offset[1] | 1,4:1]  | BRANCH |   |
| offset[12,10] | ):5] src2   | src1                  | BLT[U]  | offset[1] | 1,4:1]  | BRANCH |   |
| offset[12,10] | ):5] src2   | $\operatorname{src}1$ | BGE[U]  | offset[1  | 1,4:1]  | BRANCH |   |

#### 5) UJ-type: jal

**a.** jal rd, imm

| 31      | 30      | 21          | 20     | 19 12      | 11 7                  | 6           | 0 |
|---------|---------|-------------|--------|------------|-----------------------|-------------|---|
| imm[20] | imm[10] | :1] ir      | mm[11] | imm[19:12] | rd                    | opcode      |   |
| 1       | 10      |             | 1      | 8          | 5                     | 7           |   |
|         | o       | ffset[20:1] |        |            | $\operatorname{dest}$ | $_{ m JAL}$ |   |

| Format | Instruction | Opcode  | Funct3 | Funct6/7 |
|--------|-------------|---------|--------|----------|
|        | add         | 0110011 | 000    | 0000000  |
|        | sub         | 0110011 | 000    | 0100000  |
|        | s11         | 0110011 | 001    | 0000000  |
|        | xor         | 0110011 | 100    | 0000000  |
| R-type | srl         | 0110011 | 101    | 0000000  |
| N-type | sra         | 0110011 | 101    | 0100000  |
|        | or          | 0110011 | 110    | 0000000  |
|        | and         | 0110011 | 111    | 0000000  |
|        | 1r.d        | 0110011 | 011    | 0001000  |
|        | sc.d        | 0110011 | 011    | 0001100  |
|        | 1 b         | 0000011 | 000    | n.a.     |
|        | 1 h         | 0000011 | 001    | n.a.     |
|        | 1w          | 0000011 | 010    | n.a.     |
|        | 1 d         | 0000011 | 011    | n.a.     |
|        | 1 bu        | 0000011 | 100    | n.a.     |
|        | 1 hu        | 0000011 | 101    | n.a.     |
| I-type | 1wu         | 0000011 | 110    | n.a.     |
|        | addi        | 0010011 | 000    | n.a.     |
|        | slli        | 0010011 | 001    | 000000   |
|        | xori        | 0010011 | 100    | n.a.     |
|        | srli        | 0010011 | 101    | 000000   |
|        | srai        | 0010011 | 101    | 010000   |
|        | ori         | 0010011 | 110    | n.a.     |
|        | andi        | 0010011 | 111    | n.a.     |
|        | jalr        | 1100111 | 000    | n.a.     |

# COMP411007 Computer Architecture, Fall 2020

| Format  | Instruction | Opcode  | Funct3 | Funct6/7 |
|---------|-------------|---------|--------|----------|
|         | sb          | 0100011 | 000    | n.a.     |
| S-type  | sh          | 0100011 | 001    | n.a.     |
| 3-type  | SW          | 0100011 | 010    | n.a.     |
|         | sd          | 0100011 | 011    | n.a.     |
|         | beg         | 1100011 | 000    | n.a.     |
|         | bne         | 1100011 | 001    | n.a.     |
| SB-type | blt         | 1100011 | 100    | n.a.     |
| 3b-type | bge         | 1100011 | 101    | n.a.     |
|         | bltu        | 1100011 | 110    | n.a.     |
|         | bgeu        | 1100011 | 111    | n.a.     |
| U-type  | lui         | 0110111 | n.a.   | n.a.     |
| UJ-type | jal         | 1101111 | n.a.   | n.a.     |